# Verilog HDL: Functions & Tasks

**Pravin Zode** 

## Outline

- Functions
- Tasks
- Difference between Function and Task
- Synthesizability

# **Verilog Functions**

- Functions are designed to process inputs and return a single value
- Cannot contain delays (#), event controls (@), or nonblocking assignments (<=)</li>
- Execute in zero simulation time
- Useful for simple calculations
- Functions: Arithmetic operations like addition, multiplication

# **Verilog Function Syntax**

```
// Style 1
     function <return_type> <function_name> (<port_list>);
      return <value or expression>
 4
 5
     endfunction
 6
     // Style 2
     function <return_type> <function_name> ();
 9
       input <port list>;
10
       inout <port list>;
11
       output <port_list>;
12
13
       return <value or expression>
     endfunction
14
```

### **Function Declaration**

There are two ways to declare inputs to a function:

```
/ function [7:0] sum;
     input [7:0] a, b;
     begin
      sum = a + b;
     end
 endfunction
/ function [7:0] sum (input [7:0] a, b);
     begin
      sum = a + b;
     end
 endfunction
```

### Function Call & Return

#### **Function Return**

```
1 | sum = a + b;
```

**Function Call** 

# **Verilog Function**

```
// Gray Code Converter - Using Function
 1
 2
     module gray function (
 3
          input [3:0] bin,
 4
         output [3:0] gray
 5
 6
         function [3:0] bin2gray;
 7
              input [3:0] b;
 8
              begin
 9
                  bin2gray[3] = b[3];
10
                  bin2gray[2] = b[3] ^ b[2];
11
                  bin2gray[1] = b[2] ^ b[1];
12
                  bin2gray[0] = b[1] ^ b[0];
13
              end
          endfunction
14
15
16
          assign gray = bin2gray(bin);
17
     endmodule
```

# **Verilog Function**

```
module function_ex;
   function integer compare(input integer a, b);
   begin
       if (a > b)
5
           compare = 1;
6
7
                                                      Output
      else if (a < b)
8
           compare = -1;
                                                          10 vs 5: 1
9
       else
10
           compare = 0;
                                                          5 vs 10: -1
       end
11
                                                          7 vs 7: 0
   endfunction
13
  initial begin
14
       $display("10 vs 5: %0d", compare(10, 5));
15
       $display("5 vs 10: %0d", compare(5, 10));
16
       $display("7 vs 7: %0d", compare(7, 7));
17
18 end
19
  endmodule
```

# When Functions Are Synthesizable?

- Functions are synthesizable when they use
  - bitwise, arithmetic, and logical operators (+ ^ & | ~ << >>)
  - Conditional constructs (if-else, case, ?:)
  - Loops with fixed iteration bounds (e.g., for i=0 to7)
  - Pure combinational logic (no timing controls, delays, or events)
  - Used in always @(\*) or continuous assignments

# When Functions Are **NOT** Synthesizable?

- Functions are not synthesizable when they use
  - Unbounded / variable loops (iteration depends on runtime input)
  - Recursion (function calling itself)
  - Timing controls or delays (#, @, wait) inside functions
  - Dynamic memory / file I/O inside functions
  - ➤ Very large logic (e.g., factorial for general n) → hardware explosion

## When to Use Functions?

- Implementing reusable combinational logic (e.g., parity, Gray code, encoders)
- Improving readability by abstracting complex expressions
- Avoiding code duplication (same logic used in multiple places)
- Parameterizable small computations (bit manipulation, arithmetic helpers)
- Elaboration-time constant generation (computed once at compile time)

#### When to avoid Functions?

- Tasks requiring delays, events, or timing control → use tasks instead
- Sequential behavior with multiple clock cycles
- Large iterative computations (e.g., factorial for general
   n)
- Operations that depend on runtime iteration counts

Verilog Tasks

#### **Tasks**

- A task is a reusable procedure or subroutine in Verilog
- Can have input, output, and inout ports
- Can contain loops, conditionals, and procedural statements
- Can include timing controls (#, @) for simulation
- Useful for multi-step operations or multiple outputs
- Synthesizable only if no delays or events are used.

## Verilog Task Syntax

```
// Style 1
 2 ∨ task [name];
 3
         input [port_list];
 4
         inout [port_list];
         output [port_list];
 6
         begin
             [statements]
 8
         end
 9
     endtask
10
    // Style 2
11
12 v task [name] (input [port_list], inout [port_list], output [port_list]);
13 🗸
         begin
             [statements]
14
15
         end
16
    endtask
17
18
    // Empty port list
19 v task [name] ();
20 🗸
         begin
21
             [statements]
22
         end
23
    endtask
```

## Example: Verilog Task

```
1
     module square task(
 2
          input [3:0] num,
         output reg [7:0] result );
 4
 5
         // Task definition
 6
         task compute square;
7
              input [3:0] x;
8
              output [7:0] y;
9
              begin
10
                  y = x * x;
11
              end
12
         endtask
13
14
         // Call task in combinational block
15
         always @(*) begin
16
              compute square(num, result);
17
         end
18
     endmodule
19
```

#### Output

```
time=0 num= 3 square= 9
time=10 num= 5 square= 25
time=20 num= 8 square= 64
```

# Example: Verilog Task Reuse

```
1
     module square task reuse(
 2
          input [3:0] num1, num2, num3,
         output reg [7:0] res1, res2, res3
 3
 4
     );
 5
 6
         // Task definition
 7
         task compute square;
              input [3:0] x;
 8
 9
              output [7:0] y;
10
              begin
                  y = x * x;
11
12
              end
13
         endtask
14
15
         // Call task 3 times
16
         always @(*) begin
17
              compute_square(num1, res1);
18
              compute square(num2, res2);
19
              compute square(num3, res3);
20
         end
21
     endmodule
22
```

#### When to use Taks?

## Multiple Outputs Needed:

- Tasks allow input, output, and inout arguments.
- Useful when a single computation produces more than one result.
- Sequential or Procedural Operations: Step by-step algorithms, stimulus generation, applying test vectors
- Testbench Development: include delays (#), event controls (@), and wait statements, verification, file I/O, and simulation control
- Reusable Code Blocks: repeated procedures to avoid code duplication.

### When to avoid Taks?

- **Single Output** Only If only one result is required, prefer a function (cleaner & concise).
- Pure Combinational Expressions For simple logic (e.g., parity, Gray code, max finder), functions or assign are better.
- Inside Synthesizable RTL with TimingTasks with #, @, or wait → not synthesizable
- Overcomplication of Code Using tasks for trivial logic makes the design harder to read and maintain.
- Unnecessary Code Duplication If the same logic fits neatly in a function, tasks add extra verbosity without benefit.

# Difference Functions and Task

| Function                                                                 | Task                                                             |
|--------------------------------------------------------------------------|------------------------------------------------------------------|
| Function can enable another function but not another task                | A task can enable other tasks and functions                      |
| Function always execute in 0 simulation time                             | Task may execute in non-zero simulation time                     |
| Functions must not contain any delay, event or timing control statements | Tasks may contain delay event or timing control statements       |
| Functions must have atleast one input argument                           | Tasks may have zero or more arguments                            |
| Functions always return single value                                     | Task do not return with a value                                  |
| Functions cannot have output or inout arguments                          | Task can pass multiple values through output and inout arguments |

## Functions vs. Tasks — When to Use

#### **Use Functions When:**

- Logic is purely combinational
- Only one return value is needed
- Operation completes in a single simulation cycle
- Used for small reusable computations (e.g., parity, Gray conversion)

#### **Use Tasks When:**

- Need multiple outputs
- Timing control is required (@, #, wait)
- Sequential operations across simulation time
- More complex behavior (e.g., stimulus generation in testbenches, file I/O)

# Summary

#### **Tasks**

- Can have input, output, and inout arguments
- Can include delays, event controls, and timing constructs
- Used for complex operations or multi-step procedures
- Invoked like a procedure call: my\_task(a, b);

#### **Functions**

- Only have input arguments.
- Cannot contain delays or event controls.
- Must execute in one simulation time unit.
- Return a single value: result = my\_function(x);

Functions  $\rightarrow$  Best for RTL design (synthesizable combinational logic) Tasks  $\rightarrow$  Best for testbenches and simulation (non-synthesizable behavior)

22



Thank you!

**Happy Learning** 

23